Scrapboxのページの差分書き換えテスト (code)
code:app.js
const version="4.2.0";async function socketIO(){return(await importSocketIO())("https://scrapbox.io",{reconnectionDelay:5e3,transports:["websocket"]})}function importSocketIO(){const t=https://cdnjs.cloudflare.com/ajax/libs/socket.io/${version}/socket.io.min.js;if(document.querySelector(script[src="${t}"]))return Promise.resolve(window.io);const e=document.createElement("script");return e.src=t,new Promise((o,r)=>{e.onload=()=>o(window.io),e.onerror=n=>r(n),document.head.append(e)})}function wrap(t,e=9e4){function o(n,i){let c;return new Promise((l,d)=>{const u=s=>{clearTimeout(c),d(new Error(s))};t.emit(n,i,s=>{clearTimeout(c),t.off("disconnect",u),s.error&&d(new Error(JSON.stringify(s.error))),"data"in s?l(s?.data):l(void 0)}),c=setTimeout(()=>{t.off("disconnect",u),d(new Error(Timeout: exceeded ${e}ms))},e),t.once("disconnect",u)})}async function*r(n){let i;const c=()=>new Promise(d=>i=d),l=d=>{i?.(d)};t.on(n,l);try{for(;;)yield await c()}finally{t.off(n,l)}}return{request:o,response:r}}function toTitleLc(t){return t.replaceAll(" ","_").toLowerCase()}async function getPage(t,e){const o=await fetch(https://scrapbox.io/api/pages/${t}/${encodeURIComponent(toTitleLc(e))});if(!o.ok)throw Error(await o.text());return await o.json()}async function getUserId(){const e=await(await fetch("https://scrapbox.io/api/users/me")).json();if(e.isGuest)throw new Error("this script can only be executed by Logged in users");return e.id}function zero(t){return t.padStart(8,"0")}function createNewLineId(t){const e=Math.floor(new Date().getTime()/1e3).toString(16),o=Math.floor(16777214*Math.random()).toString(16);return${zero(e).slice(-8)}${t.slice(-6)}0000${zero(o)}}async function getPageIdAndCommitId(t,e){const{id:o,commitId:r,persistent:n}=await getPage(t,e);return{pageId:o,commitId:r,persistent:n}}async function getProjectId(t){return(await(await fetch(https://scrapbox.io/api/projects/${t})).json()).id}function diff(t,e){const o=t.length>e.length,r=o?e:t,n=o?t:e,i=r.length+1,c=r.length+n.length+3,l=new Array(c);l.fill(-1);const d=[];function u(a,h,g){let y=Math.max(h,g),x=y-a;for(;x<r.length&&y<n.length&&rx===ny;)++x,++y;return la+i=d.length,d.push([{x,y},la+(h>g?-1:1)+i]),y}const s=new Array(c);s.fill(-1);let w=-1;const f=n.length-r.length;do{++w;for(let a=-w;a<=f-1;++a)sa+i=u(a,sa-1+i+1,sa+1+i);for(let a=f+w;a>=f+1;--a)sa+i=u(a,sa-1+i+1,sa+1+i);sf+i=u(f,sf-1+i+1,sf+1+i)}while(sf+i!==n.length);const m=[];let p=lf+i;for(;p!==-1;)m.push(dp0),p=dp1;return{from:t,to:e,editDistance:f+w*2,buildSES:function*(){let a=0,h=0;for(const{x:g,y}of reverse(m))for(;a<g||h<y;)y-g>h-a?(yield{value:nh,type:o?"deleted":"added"},++h):y-g<h-a?(yield{value:ra,type:o?"added":"deleted"},++a):(yield{value:ra,type:"common"},++a,++h)}}}function*toExtendedChanges(t){let e=[],o=[];function*r(){if(e.length>o.length){for(let n=0;n<o.length;n++)yield makeReplaced(en,on);for(let n=o.length;n<e.length;n++)yield en}else{for(let n=0;n<e.length;n++)yield makeReplaced(en,on);for(let n=e.length;n<o.length;n++)yield on}e=[],o=[]}for(const n of t)switch(n.type){case"added":e.push(n);break;case"deleted":o.push(n);break;case"common":yield*r(),yield n;break}yield*r()}function makeReplaced(t,e){return{value:t.value,oldValue:e.value,type:"replaced"}}function*reverse(t){for(let e=t.length-1;e>=0;e--)yield te}function*diffToChanges(t,e,{userId:o}){const{buildSES:r}=diff(t.map(({text:c})=>c),e);let n=0,i=t0.id;for(const c of toExtendedChanges(r())){switch(c.type){case"added":yield{_insert:i,lines:{id:createNewLineId(o),text:c.value}};continue;case"deleted":yield{_delete:i,lines:-1};break;case"replaced":yield{_update:i,lines:{text:c.value}};break}n++,i=tn?.id??"_end"}}async function joinPageRoom(t,e){const{pageId:o,commitId:r,persistent:n},i,c=await Promise.all(getPageIdAndCommitId(t,e),getProjectId(t),getUserId());let l=r,d=n;const u=await socketIO(),{request:s,response:w}=wrap(u);await s("socket.io-request",{method:"room:join",data:{projectId:i,pageId:o,projectUpdatesStream:!1}}),(async()=>{for await(const{id:m}of w("commit"))l=m})();async function f(m,p=3){d||(l=(await pushCommit(s,{title:e},{parentId:l,projectId:i,pageId:o,userId:c})).commitId,d=!0),l=await pushWithRetry(s,m,{parentId:l,projectId:i,pageId:o,userId:c,project:t,title:e,retry:p})}return{insert:async(m,p="_end")=>{const a=m.split(/\n|\r\n/).map(h=>({_insert:p,lines:{text:h,id:createNewLineId(c)}}));await f(a)},remove:m=>f({_delete:m,lines:-1}),update:(m,p)=>f({_update:p,lines:{text:m}}),patch:async m=>{const p=(await getPage(t,e)).lines,a=m(p),h=...diffToChanges(p,a,{userId:c});await f(h)},listenPageUpdate:()=>w("commit"),cleanup:()=>u.disconnect()}}async function pushCommit(t,e,{projectId:o,pageId:r,userId:n,parentId:i}){return e.length===0?{commitId:i}:await t("socket.io-request",{method:"commit",data:{kind:"page",projectId:o,parentId:i,pageId:r,userId:n,changes:e,cursor:null,freeze:!0}})}async function pushWithRetry(t,e,{projectId:o,pageId:r,userId:n,parentId:i,project:c,title:l,retry:d=3}){try{i=(await pushCommit(t,e,{parentId:i,projectId:o,pageId:r,userId:n})).commitId}catch(u){console.log("Faild to push a commit. Retry after pulling new commits");for(let s=0;s<d;s++)try{i=(await getPageIdAndCommitId(c,l)).commitId,i=(await pushCommit(t,e,{parentId:i,projectId:o,pageId:r,userId:n})).commitId,console.log("Success in retrying");break}catch(w){continue}throw Error("Faild to retry pushing.")}return i}function throttle(t,e){const{trailing:o=!1}=e??{};let r,n=!1;const i=()=>r?.resolve?.({executed:!1}),c=async()=>{if(n||!r)return;n=!0;const{parameters:l,resolve:d,reject:u}=r;r=void 0;try{const s=await t(...l);n=!1,d({result:s,executed:!0})}catch(s){n=!1,u(s)}finally{o?await c():(i(),await Promise.resolve())}};return(...l)=>new Promise((d,u)=>{r?.resolve?.({executed:!1}),r={parameters:l,resolve:d,reject:u},c()})}const project="takker",title="\u5DEE\u5206\u66F8\u304D\u8FBC\u307F\u30C6\u30B9\u30C8\u7528\u30DA\u30FC\u30B8",div=document.createElement("div"),shadowRoot=div.attachShadow({mode:"open"});shadowRoot.innerHTML=` <style>
:host {
position: fixed;
left: 0;
bottom: 0;
z-index: 9999;
text-decoration: none;
background-color: var(--page-bg);
color: var(--page-text-color);
border: solid 1px var(--telomere-unread);
border-radius: 4px;
padding: 8px;
margin: 10px;
}
button {
display: inline-block;
}
</style>
<div>
<button id="button">x</button>
<span id="title"></span>
</div>
<textarea id="editor"></textarea>
,document.body.append(div);const{patch,cleanup}=await joinPageRoom(project,title);console.info("Connected to scrapbox.io");const editor=shadowRoot.getElementById("editor"),update=t=>patch(([e])=>[e.text,...t.split(/\n|\r\n/)]),callback=throttle(update,{trailing:!0});editor.addEventListener("input",()=>callback(editor.value)),console.info("Start patching"),shadowRoot.getElementById("title").insertAdjacentHTML("beforeend",<a href="${/${project}/${title}}" target="_blank">${/${project}/${title}}</a>`),shadowRoot.getElementById("button").onclick=()=>{console.info("End patching"),cleanup(),div.remove()};
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["<stdin>"],
  "sourcesContent": ["const version = \"4.2.0\";\nasync function socketIO() {\n    const io = await importSocketIO();\n    return io(\"https://scrapbox.io\", {\n        reconnectionDelay: 5000,\n        transports: [\n            \"websocket\"\n        ]\n    });\n}\nfunction importSocketIO() {\n    const url = `https://cdnjs.cloudflare.com/ajax/libs/socket.io/${version}/socket.io.min.js`;\n    if (document.querySelector(`script[src=\"${url}\"]`)) {\n        return Promise.resolve(window.io);\n    }\n    const script = document.createElement(\"script\");\n    script.src = url;\n    return new Promise((resolve, reject)=>{\n        script.onload = ()=>resolve(window.io)\n        ;\n        script.onerror = (e)=>reject(e)\n        ;\n        document.head.append(script);\n    });\n}\nfunction wrap(socket, timeout = 90000) {\n    function request(event, data) {\n        let id;\n        return new Promise((resolve, reject)=>{\n            const onDisconnect = (message)=>{\n                clearTimeout(id);\n                reject(new Error(message));\n            };\n            socket.emit(event, data, (response)=>{\n                clearTimeout(id);\n                socket.off(\"disconnect\", onDisconnect);\n                if (response.error) {\n                    reject(new Error(JSON.stringify(response.error)));\n                }\n                if (\"data\" in response) {\n                    resolve(response?.data);\n                } else {\n                    resolve(undefined);\n                }\n            });\n            id = setTimeout(()=>{\n                socket.off(\"disconnect\", onDisconnect);\n                reject(new Error(`Timeout: exceeded ${timeout}ms`));\n            }, timeout);\n            socket.once(\"disconnect\", onDisconnect);\n        });\n    }\n    async function* response(event) {\n        let _resolve;\n        const waitForEvent = ()=>new Promise((res)=>_resolve = res\n            )\n        ;\n        const resolve = (data)=>{\n            _resolve?.(data);\n        };\n        socket.on(event, resolve);\n        try {\n            while(true){\n                yield await waitForEvent();\n            }\n        } finally{\n            socket.off(event, resolve);\n        }\n    }\n    return {\n        request,\n        response\n    };\n}\nfunction toTitleLc(title) {\n    return title.replaceAll(\" \", \"_\").toLowerCase();\n}\nasync function getPage(project, title) {\n    const res = await fetch(`https://scrapbox.io/api/pages/${project}/${encodeURIComponent(toTitleLc(title))}`);\n    if (!res.ok) throw Error(await res.text());\n    return await res.json();\n}\nasync function getUserId() {\n    const res = await fetch(\"https://scrapbox.io/api/users/me\");\n    const json = await res.json();\n    if (json.isGuest) {\n        throw new Error(\"this script can only be executed by Logged in users\");\n    }\n    return json.id;\n}\nfunction zero(n) {\n    return n.padStart(8, \"0\");\n}\nfunction createNewLineId(userId) {\n    const time = Math.floor(new Date().getTime() / 1000).toString(16);\n    const rand = Math.floor(16777214 * Math.random()).toString(16);\n    return `${zero(time).slice(-8)}${userId.slice(-6)}0000${zero(rand)}`;\n}\nasync function getPageIdAndCommitId(project, title) {\n    const { id , commitId , persistent  } = await getPage(project, title);\n    return {\n        pageId: id,\n        commitId,\n        persistent\n    };\n}\nasync function getProjectId(project) {\n    const res = await fetch(`https://scrapbox.io/api/projects/${project}`);\n    const json = await res.json();\n    return json.id;\n}\nfunction diff(left, right) {\n    const reversed = left.length > right.length;\n    const a = reversed ? right : left;\n    const b = reversed ? left : right;\n    const offset = a.length + 1;\n    const MAXSIZE = a.length + b.length + 3;\n    const path = new Array(MAXSIZE);\n    path.fill(-1);\n    const pathpos = [];\n    function snake(k, p, pp) {\n        let y = Math.max(p, pp);\n        let x = y - k;\n        while(x < a.length && y < b.length && a[x] === b[y]){\n            ++x;\n            ++y;\n        }\n        path[k + offset] = pathpos.length;\n        pathpos.push([\n            {\n                x,\n                y\n            },\n            path[k + (p > pp ? -1 : +1) + offset]\n        ]);\n        return y;\n    }\n    const fp = new Array(MAXSIZE);\n    fp.fill(-1);\n    let p = -1;\n    const delta = b.length - a.length;\n    do {\n        ++p;\n        for(let k = -p; k <= delta - 1; ++k){\n            fp[k + offset] = snake(k, fp[k - 1 + offset] + 1, fp[k + 1 + offset]);\n        }\n        for(let k1 = delta + p; k1 >= delta + 1; --k1){\n            fp[k1 + offset] = snake(k1, fp[k1 - 1 + offset] + 1, fp[k1 + 1 + offset]);\n        }\n        fp[delta + offset] = snake(delta, fp[delta - 1 + offset] + 1, fp[delta + 1 + offset]);\n    }while (fp[delta + offset] !== b.length)\n    const epc = [];\n    let r = path[delta + offset];\n    while(r !== -1){\n        epc.push(pathpos[r][0]);\n        r = pathpos[r][1];\n    }\n    return {\n        from: left,\n        to: right,\n        editDistance: delta + p * 2,\n        buildSES: function*() {\n            let xIndex = 0;\n            let yIndex = 0;\n            for (const { x , y  } of reverse(epc)){\n                while(xIndex < x || yIndex < y){\n                    if (y - x > yIndex - xIndex) {\n                        yield {\n                            value: b[yIndex],\n                            type: reversed ? \"deleted\" : \"added\"\n                        };\n                        ++yIndex;\n                    } else if (y - x < yIndex - xIndex) {\n                        yield {\n                            value: a[xIndex],\n                            type: reversed ? \"added\" : \"deleted\"\n                        };\n                        ++xIndex;\n                    } else {\n                        yield {\n                            value: a[xIndex],\n                            type: \"common\"\n                        };\n                        ++xIndex;\n                        ++yIndex;\n                    }\n                }\n            }\n        }\n    };\n}\nfunction* toExtendedChanges(changes) {\n    let addedList = [];\n    let deletedList = [];\n    function* flush() {\n        if (addedList.length > deletedList.length) {\n            for(let i = 0; i < deletedList.length; i++){\n                yield makeReplaced(addedList[i], deletedList[i]);\n            }\n            for(let i1 = deletedList.length; i1 < addedList.length; i1++){\n                yield addedList[i1];\n            }\n        } else {\n            for(let i = 0; i < addedList.length; i++){\n                yield makeReplaced(addedList[i], deletedList[i]);\n            }\n            for(let i1 = addedList.length; i1 < deletedList.length; i1++){\n                yield deletedList[i1];\n            }\n        }\n        addedList = [];\n        deletedList = [];\n    }\n    for (const change of changes){\n        switch(change.type){\n            case \"added\":\n                addedList.push(change);\n                break;\n            case \"deleted\":\n                deletedList.push(change);\n                break;\n            case \"common\":\n                yield* flush();\n                yield change;\n                break;\n        }\n    }\n    yield* flush();\n}\nfunction makeReplaced(left, right) {\n    return {\n        value: left.value,\n        oldValue: right.value,\n        type: \"replaced\"\n    };\n}\nfunction* reverse(list) {\n    for(let i = list.length - 1; i >= 0; i--){\n        yield list[i];\n    }\n}\nfunction* diffToChanges(left, right, { userId  }) {\n    const { buildSES  } = diff(left.map(({ text  })=>text\n    ), right);\n    let lineNo = 0;\n    let lineId = left[0].id;\n    for (const change of toExtendedChanges(buildSES())){\n        switch(change.type){\n            case \"added\":\n                yield {\n                    _insert: lineId,\n                    lines: {\n                        id: createNewLineId(userId),\n                        text: change.value\n                    }\n                };\n                continue;\n            case \"deleted\":\n                yield {\n                    _delete: lineId,\n                    lines: -1\n                };\n                break;\n            case \"replaced\":\n                yield {\n                    _update: lineId,\n                    lines: {\n                        text: change.value\n                    }\n                };\n                break;\n        }\n        lineNo++;\n        lineId = left[lineNo]?.id ?? \"_end\";\n    }\n}\nasync function joinPageRoom(project, title) {\n    const [{ pageId , commitId: initialCommitId , persistent  }, projectId, userId] = await Promise.all([\n        getPageIdAndCommitId(project, title),\n        getProjectId(project),\n        getUserId(), \n    ]);\n    let parentId = initialCommitId;\n    let created = persistent;\n    const io = await socketIO();\n    const { request , response  } = wrap(io);\n    await request(\"socket.io-request\", {\n        method: \"room:join\",\n        data: {\n            projectId,\n            pageId,\n            projectUpdatesStream: false\n        }\n    });\n    (async ()=>{\n        for await (const { id  } of response(\"commit\")){\n            parentId = id;\n        }\n    })();\n    async function push(changes, retry = 3) {\n        if (!created) {\n            const res = await pushCommit(request, [\n                {\n                    title\n                }\n            ], {\n                parentId,\n                projectId,\n                pageId,\n                userId\n            });\n            parentId = res.commitId;\n            created = true;\n        }\n        parentId = await pushWithRetry(request, changes, {\n            parentId,\n            projectId,\n            pageId,\n            userId,\n            project,\n            title,\n            retry\n        });\n    }\n    return {\n        insert: async (text, beforeId = \"_end\")=>{\n            const changes = text.split(/\\n|\\r\\n/).map((line)=>({\n                    _insert: beforeId,\n                    lines: {\n                        text: line,\n                        id: createNewLineId(userId)\n                    }\n                })\n            );\n            await push(changes);\n        },\n        remove: (lineId)=>push([\n                {\n                    _delete: lineId,\n                    lines: -1\n                }\n            ])\n        ,\n        update: (text, lineId)=>push([\n                {\n                    _update: lineId,\n                    lines: {\n                        text\n                    }\n                }\n            ])\n        ,\n        patch: async (update)=>{\n            const oldLines = (await getPage(project, title)).lines;\n            const newLines = update(oldLines);\n            const changes = [\n                ...diffToChanges(oldLines, newLines, {\n                    userId\n                })\n            ];\n            await push(changes);\n        },\n        listenPageUpdate: ()=>response(\"commit\")\n        ,\n        cleanup: ()=>io.disconnect()\n    };\n}\nasync function pushCommit(request, changes, { projectId , pageId , userId , parentId  }) {\n    if (changes.length === 0) return {\n        commitId: parentId\n    };\n    const res = await request(\"socket.io-request\", {\n        method: \"commit\",\n        data: {\n            kind: \"page\",\n            projectId,\n            parentId,\n            pageId,\n            userId,\n            changes,\n            cursor: null,\n            freeze: true\n        }\n    });\n    return res;\n}\nasync function pushWithRetry(request, changes, { projectId , pageId , userId , parentId , project , title , retry =3  }) {\n    try {\n        const res = await pushCommit(request, changes, {\n            parentId,\n            projectId,\n            pageId,\n            userId\n        });\n        parentId = res.commitId;\n    } catch (_e) {\n        console.log(\"Faild to push a commit. Retry after pulling new commits\");\n        for(let i = 0; i < retry; i++){\n            try {\n                parentId = (await getPageIdAndCommitId(project, title)).commitId;\n                const res = await pushCommit(request, changes, {\n                    parentId,\n                    projectId,\n                    pageId,\n                    userId\n                });\n                parentId = res.commitId;\n                console.log(\"Success in retrying\");\n                break;\n            } catch (_e) {\n                continue;\n            }\n        }\n        throw Error(\"Faild to retry pushing.\");\n    }\n    return parentId;\n}\nfunction throttle(callback, options) {\n    const { trailing =false  } = options ?? {\n    };\n    let queue;\n    let running = false;\n    const cancel = ()=>queue?.resolve?.({\n            executed: false\n        })\n    ;\n    const runNext = async ()=>{\n        if (running || !queue) {\n            return;\n        }\n        running = true;\n        const { parameters , resolve , reject  } = queue;\n        queue = undefined;\n        try {\n            const result = await callback(...parameters);\n            running = false;\n            resolve({\n                result,\n                executed: true\n            });\n        } catch (e) {\n            running = false;\n            reject(e);\n        } finally{\n            if (trailing) {\n                await runNext();\n            } else {\n                cancel();\n                await Promise.resolve();\n            }\n        }\n    };\n    return (...parameters)=>new Promise((resolve, reject)=>{\n            queue?.resolve?.({\n                executed: false\n            });\n            queue = {\n                parameters,\n                resolve,\n                reject\n            };\n            runNext();\n        })\n    ;\n}\nconst project = \"takker\";\nconst title = \"\u5DEE\u5206\u66F8\u304D\u8FBC\u307F\u30C6\u30B9\u30C8\u7528\u30DA\u30FC\u30B8\";\nconst div = document.createElement(\"div\");\nconst shadowRoot = div.attachShadow({\n    mode: \"open\"\n});\nshadowRoot.innerHTML = `\n  <style>\n    :host {\n       position: fixed;\n       left: 0;\n       bottom: 0;\n       z-index: 9999;\n       text-decoration: none;\n       background-color: var(--page-bg);\n       color: var(--page-text-color);\n       border: solid 1px var(--telomere-unread);\n       border-radius: 4px;\n       padding: 8px;\n       margin: 10px;\n    }\n    button {\n     display: inline-block;\n   }\n  </style>\n  <div>\n    <button id=\"button\">x</button>\n    <span id=\"title\"></span>\n  </div>\n  <textarea id=\"editor\"></textarea>\n`;\ndocument.body.append(div);\nconst { patch , cleanup  } = await joinPageRoom(project, title);\nconsole.info(\"Connected to scrapbox.io\");\nconst editor = shadowRoot.getElementById(\"editor\");\nconst update = (text)=>patch(([titleLine])=>[\n            titleLine.text,\n            ...text.split(/\\n|\\r\\n/)\n        ]\n    )\n;\nconst callback = throttle(update, {\n    trailing: true\n});\neditor.addEventListener(\"input\", ()=>callback(editor.value)\n);\nconsole.info(\"Start patching\");\nshadowRoot.getElementById(\"title\").insertAdjacentHTML(\"beforeend\", `<a href=\"${`/${project}/${title}`}\" target=\"_blank\">${`/${project}/${title}`}</a>`);\nshadowRoot.getElementById(\"button\").onclick = ()=>{\n    console.info(\"End patching\");\n    cleanup();\n    div.remove();\n};\n\n"],
  "mappings": "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",
  "names": []
}
